Problem: Is there a good way to improve performance under concurrent requests from a single client in ASP.NET MVC when using writable session variables?
Currently, as of version 5.1, ASP.NET MVC will queue concurrent requests from a single client when writable (i.e. not read-only) session variables are used (see The Downsides of ASP.NET Session State). This queuing occurs because MVC locks session variables to ensure thread safety. Unfortunately, the lock is implemented such that the session is only polled every 500ms to check if it is free (see Storing Anything in ASP.NET Session Causes 500ms Delays). As a result, performance can be very poor when multiple requests are made simultaneously from the same client through means such as asynchronous AJAX requests.
To avoid queuing, session can be turned off or set to read-only for specific controllers—possibly automatically if not otherwise specified (see SessionStateBehavior Enumeration). In some cases, allowing a controller to write to session may be desirable or necessary. If this is the case, there are at least two ways that performance may be increased without forgoing the use of writeable session variables. Both of these solutions that we will describe will work on a project-wide scale and will not require session variables to be used differently outside of the thread safety considerations related to Solution 2.
Solution 1: Decrease the Lock Polling Interval
Because the polling interval of the session lock is defined as a static constant within the SessionStateModule class (Microsoft .NET Framework 4.6.2 Reference Source), it may be modified through the use of reflection (Storing Anything in ASP.NET Session Causes 500ms Delays). LOCKED_ITEM_POLLING_INTERVAL is the interval at which polling is performed with a default value of 500ms. LOCKED_ITEM_POLLING_DELTA is the minimum time between polling checks with a default value of 250ms. Code to change these values is provided in Figure 1 below. It may be placed in the project’s Global.asax file, though it isn’t required that it should be placing there.
1 2 3 4 5 |
var sessionStateModuleType = typeof(SessionStateModule); var pollingIntervalFieldInfo = sessionStateModuleType.GetField("LOCKED_ITEM_POLLING_INTERVAL", BindingFlags.NonPublic | BindingFlags.Static); pollingIntervalFieldInfo.SetValue(null, 30); // default 500ms var pollingDeltaFieldInfo = sessionStateModuleType.GetField("LOCKED_ITEM_POLLING_DELTA", BindingFlags.NonPublic | BindingFlags.Static); pollingDeltaFieldInfo.SetValue(null, TimeSpan.FromMilliseconds(15.0)); // default 250ms |
The particular values used here are somewhat arbitrary; however, in general, values closer in duration to the length of time it takes a typical request to be processed are likely desirable. Setting the values too low may lead to an undesirably high usage of CPU resources. Experimentation may need to be performed if performance is initially unsatisfactory.
Although requests will still queue after these modifications are made, performance may increase significantly, especially when many quickly processed requests are made simultaneously.
Solution 2: Implement a Lockless SessionStateStoreProvider
MVC allows for the use of custom session state store providers through the inheritance of the SessionStateStoreProviderBase class (MSDN: Implementing a Session-State Store Provider, MSDN: Session-State Modes). A simple way to use this to our advantage is to use reflection to wrap the pre-existing in-memory session state store provider, InProcSessionStateStore
, in a simple class that uses calls to InProcSessionStateStore
methods for most of its method implementations. When we encounter locks in either of the methods that are used to retrieve a session, GetItem
and GetItemExclusive
, we immediately release them using ReleaseItemExclusive
and then again attempt to retrieve the now-unlocked session. As a result, session becomes effectively lockless. An implementation of the described class is provided in Figure 2 below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 |
public class LocklessInProcSessionStateStore : SessionStateStoreProviderBase { private SessionStateStoreProviderBase _store; public override void Initialize(string name, NameValueCollection config) { base.Initialize(name, config); var storeType = typeof(SessionStateStoreProviderBase).Assembly.GetType("System.Web.SessionState.InProcSessionStateStore"); _store = (SessionStateStoreProviderBase)Activator.CreateInstance(storeType); _store.Initialize(name, config); } public override void Dispose() { _store.Dispose(); } public override bool SetItemExpireCallback(SessionStateItemExpireCallback expireCallback) { return _store.SetItemExpireCallback(expireCallback); } public override void InitializeRequest(HttpContext context) { _store.InitializeRequest(context); } public override SessionStateStoreData GetItem(HttpContext context, string id, out bool locked, out TimeSpan lockAge, out object lockId, out SessionStateActions actions) { var returnValue = _store.GetItem(context, id, out locked, out lockAge, out lockId, out actions); if (returnValue == null && lockId != null) { _store.ReleaseItemExclusive(context, id, lockId); returnValue = _store.GetItem(context, id, out locked, out lockAge, out lockId, out actions); } return returnValue; } public override SessionStateStoreData GetItemExclusive(HttpContext context, string id, out bool locked, out TimeSpan lockAge, out object lockId, out SessionStateActions actions) { var returnValue = _store.GetItemExclusive(context, id, out locked, out lockAge, out lockId, out actions); if (returnValue == null && lockId != null) { _store.ReleaseItemExclusive(context, id, lockId); returnValue = _store.GetItemExclusive(context, id, out locked, out lockAge, out lockId, out actions); } return returnValue; } public override void ReleaseItemExclusive(HttpContext context, string id, object lockId) { _store.ReleaseItemExclusive(context, id, lockId); } public override void SetAndReleaseItemExclusive(HttpContext context, string id, SessionStateStoreData item, object lockId, bool newItem) { _store.SetAndReleaseItemExclusive(context, id, item, lockId, newItem); } public override void RemoveItem(HttpContext context, string id, object lockId, SessionStateStoreData item) { _store.RemoveItem(context, id, lockId, item); } public override void ResetItemTimeout(HttpContext context, string id) { _store.ResetItemTimeout(context, id); } public override SessionStateStoreData CreateNewStoreData(HttpContext context, int timeout) { return _store.CreateNewStoreData(context, timeout); } public override void CreateUninitializedItem(HttpContext context, string id, int timeout) { _store.CreateUninitializedItem(context, id, timeout); } public override void EndRequest(HttpContext context) { _store.EndRequest(context); } } |
Additionally, the XML in Figure 3 below must be placed in the system.web element of the Web.config file of the project in order for the custom session state store provider to be used. The “type” attribute of the LocklessInProcSessionStateStore
provider should be the fully qualified class name of the wrapper class. The values provided for the “cookieless” and “timeout” attributes may be changed.
1 2 3 4 5 6 |
<sessionState mode="Custom" customProvider="LocklessInProcSessionStateStore" cookieless="false" timeout="1" > <providers> <add name="LocklessInProcSessionStateStore" type="IPMS.IPManager.IDS.Server.Web.Session.LocklessInProcSessionStateStore"/> </providers> </sessionState> |
This solution will allow for even better performance than that provided by Solution 1; however, it gives up thread safety. If this solution is to be used, each use of session will need to be carefully considered in order to avoid typical multi-threading pitfalls such as race conditions. As such, if it is implemented in an existing project, it may introduce subtle and not-so-subtle bugs.
Example Performance Comparison:
Below are some images of the network tab in Google Chrome’s developer console in Figures 4, 5, and 6 as examples of the behavior exhibited by implementations with No Solution, Solution 1, and Solution 2 on a page that performs 7 concurrent asynchronous AJAX requests upon loading. The small green bars on the right side of each image represent active client-side AJAX requests over time. The images show an approximately common horizontal time scale. Note that the scales are different because the latency of the standard session is so high. the actual performance differences are highly circumstantial and variable in nature. Both solutions provide a dramatic improvement, but the first solution, Decreasing the Lock Polling Interval, is a safer solution at a minor cost in performance.
Figure 4. Concurrent Request Performance for No Solution (standard session)
Load comments